from flask import abort
from flask import current_app, jsonify
from flask import g
from flask import render_template
from flask import request
from flask import session

from info import db
from info.models import News, User, Comment, CommentLike
from info.utils.commons import user_login_data
from info.utils.response_code import RET
from . import  news_blue

#####5. 关注与取消关注
# 请求路径: /news/followed_user
# 请求方式: POST
# 请求参数:user_id,action
# 返回值: errno, errmsg

@news_blue.route('/followed_user', methods=['POST'])
@user_login_data
def followed_user():
    '''
    0.判断用户是否登陆
    1.获取参数
    2.校验参数,为空校验
    3.校验操作类型
    4.根据作者的编号取出作者对象,判断作者    对象是否存在
    5.根据操作类型,进行关注&取消关注
    6.返回响应
    :return:
    '''
    # 0.判断用户是否登陆
    if not g.user:
        return jsonify(errno=RET.NODATA,errmsg='用户未登陆')
    # 1.获取参数
    author_id = request.json.get('user_id')
    action = request.json.get('action')
    # 2.校验参数,为空校验
    if not all([author_id,action]):
        return jsonify(errno=RET.PARAMERR,errmsg='参数不全')
    # 3.校验操作类型
    if not action in ['follow','unfollow']:
        return jsonify(errno=RET.DATAERR,errmsg='操作类型有误')
    # 4.根据作者的编号取出作者对象,判断作者对象是否存在
    try:
        author = User.query.get(author_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='获取作者失败')
    if not author: return jsonify(errno=RET.NODATA,errmsg='该作者不存在')

    # 5.根据操作类型,进行关注&取消关注
    if action == 'follow':
        #5.1 判断当前的用户是否关注了该作者
        if not g.user in author.followers:
            author.followers.append(g.user)
    else:
        #5.2 判断当前的用户是否关注了作者
        if g.user in author.followers:
            author.followers.remove(g.user)
    # 6.返回响应
    return jsonify(errno=RET.OK,errmsg='操作成功')






####  评论点赞

# 请求路径: /news/comment_like
# 请求方式: POST
# 请求参数:news_id,comment_id,action,g.user
# 返回值: errno,errmsg

@news_blue.route('/comment_like', methods=['GET', 'POST'])
@user_login_data
def comment_like():

    """
      1. 判断用户是否有登陆
      2. 获取参数
      3. 参数校验,为空校验
      4. 操作类型进行校验
      5. 通过评论编号查询评论对象,并判断是否存在
      6. 根据操作类型点赞取消点赞
      7. 返回响应
      :return:
      """
    # 1. 判断用户是否有登陆
    if not g.user:
        return jsonify(errno=RET.NODATA,errmsg='用户未登陆')
    # 2. 获取参数
    comment_id = request.json.get('comment_id')
    action = request.json.get('action')

    # 3. 参数校验,
    if not all([comment_id,action]):
        return jsonify(errno=RET.PARAMERR,errmsg='参数不全')
    # 4. 操作类型进行校验
    if not action in ['add','remove']:
        return jsonify(errno=RET.DATAERR,errmsg='操作类型有误')

    # 5. 通过评论编号查询评论对象,并判断是否存在
    try:
        comment = Comment.query.get(comment_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='获取评论失败')

    if not comment: return jsonify(errno=RET.NODATA,errmsg='评论不存在')

    # 6. 根据操作类型点赞取消点赞
    try:
        if action == 'add':
            #6.1 判断用户是否对该评论点过赞
            comment_like = CommentLike.query.filter(CommentLike.user_id == g.user.id,CommentLike.comment_id == comment_id).first()
            if not comment_like:
                #创建点赞对象
                comment_like = CommentLike()
                comment_like.user_id = g.user.id
                comment_like.comment_id = comment_id

                #添加到数据库中
                db.session.add(comment_like)


                #将该评论的点赞数量+1
                comment.like_count += 1
                db.session.commit()

        else:
            # 6.2 判断用户是否对该评论点过赞
            comment_like = CommentLike.query.filter(CommentLike.user_id == g.user.id,CommentLike.comment_id == comment_id).first()
            if comment_like:
                # 删除点赞对象
               db.session.delete(comment_like)

                # 将该评论的点赞数量-1
            if comment.like_count > 0:

                comment.like_count -= 1
            db.session.commit()

    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='操作失败')

    # 7. 返回响应
    return jsonify(errno=RET.OK,errmsg='操作成功')



### 新闻评论后端

# 请求路径: /news/news_comment
# 请求方式: POST
# 请求参数:news_id,comment,parent_id, g.user
# 返回值: errno,errmsg,评论字典
@news_blue.route('/news_comment', methods=['POST'])
@user_login_data
def news_comment():
    """
    1. 判断用户是否登陆
    2. 获取请求参数
    3. 校验参数,为空校验
    4. 根据新闻编号取出新闻对象,判断新闻是否存在
    5. 创建评论对象,设置属性
    6. 保存评论对象到数据库中
    7. 返回响应,携带评论的数据
    :return:
    """
    # 1. 判断用户是否登陆
    if not g.user:
        return jsonify(errno=RET.NODATA,errmsg='用户未登录')
    # 2. 获取请求参数
    news_id = request.json.get('news_id')
    content = request.json.get('comment')
    parent_id = request.json.get('parent_id')

    # 3. 校验参数,为空校验
    if not all([news_id,content]):
        return jsonify(errno=RET.PARAMERR,errmsg='参数不全')
    # 4. 根据新闻编号取出新闻对象,判断新闻是否存在
    try:
        news = News.query.get(news_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='获取新闻失败')
    if not news: return jsonify(errno=RET.NODATA,errmsg='新闻不存在')

    # 5. 创建评论对象,设置属性
    comment = Comment()
    comment.user_id = g.user.id
    comment.news_id = news_id
    comment.content = content
    if parent_id:
        comment.parent_id = parent_id

    # 6. 保存评论对象到数据库中
    try:
        db.session.add(comment)
        db.session.commit()
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='评论失败')

    # 7. 返回响应,携带评论的数据

    return jsonify(errno=RET.OK,errmsg='评论成功',data = comment.to_dict())



## 收藏功能接口

# 请求路径: /news/news_collect
# 请求方式: POST
# 请求参数:news_id,action, g.user
# 返回值: errno,errmsg

@news_blue.route('/news_collect', methods=['POST'])
@user_login_data
def news_collect():

    """
     1. 判断用户是否登陆了
     2. 获取参数
     3. 参数校验,为空校验
     4. 操作类型校验
     5. 根据新闻的编号取出新闻对象
     6. 判断新闻对象是否存在
     7. 根据操作类型,进行收藏&取消收藏操作
     8. 返回响应
     :return:
     """
    # 1. 判断用户是否登陆了
    if not g.user:
        return jsonify(errno=RET.NODATA,errmsg='用户未登录')
    # 2. 获取参数
    news_id = request.json.get('news_id')
    action = request.json.get('action')

    # 3. 参数校验,为空校验
    if not all([news_id,action]):
        return jsonify(errno=RET.PARAMERR,errmsg='参数不全')
    # 4. 操作类型校验
    if not action in ['collect','cancel_collect']:
        return jsonify(errno=RET.DATAERR,errmsg='操作类型有误')
    # 5. 根据新闻的编号取出新闻对象
    try:
        news = News.query.get(news_id)
    except Exception as e:
       current_app.logger.error(e)
       return jsonify(errno=RET.DBERR,errmsg='新闻获取失败')

    # 6. 判断新闻对象是否存在
    if not news:
        return jsonify(errno=RET.NODATA,errmsg='新闻不存在')

    # 7. 根据操作类型,进行收藏&取消收藏操作
    if action == "collect":
        #7.1 判断用户是否有对该新闻做过收藏
        if not news in g.user.collection_news:
            g.user.collection_news.append(news)
    else:
        #7.2判断用户是否有对该新闻做过收藏
        if news in g.user.collection_news:
            g.user.collection_news.remove(news)

    # 8. 返回响应
    return jsonify(errno=RET.OK,errmsg='操作成功')






#渲染新闻页面

# 请求路径: /news/<int:news_id>
# 请求方式: GET
# 请求参数:news_id
# 返回值: detail.html页面, 用户data字典数据
@news_blue.route('/<int:news_id>')
@user_login_data
def news_detail(news_id):

    # #0.从session中取出用户的user_id
    # user_id = session.get('user_id')
    #
    # #0.1 通过user_id取出用户对象
    # user = None
    # if user_id:
    #     try:
    #         user = User.query.get(user_id)
    #     except Exception as e:
    #         current_app.logger.error(e)


    #1.根据新闻编号,查询新闻对象
    try:
        news = News.query.get(news_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='获取新闻失败')

    #2.如果新闻对象不存在直接抛出异常
    if not news:
        abort(404)

    #3.获取前6条新闻
    try:
        click_news = News.query.order_by(News.clicks.desc()).limit(6).all()
    except Exception as e:
        current_app.logger.error(e)
    #4.将热门新闻的对象列表,转成字典列表
    click_news_list = []
    for item in click_news:
        click_news_list.append(item.to_dict())

    #5.判断用户是否收藏过该新闻
    is_collected = False
    #用户需要登陆,并且该新闻在用户收藏的新闻列表中
    if g.user:
        if news in g.user.collection_news:
            is_collected = True

    #6.查询数据库中，该新闻的所有内容
    try:
        comments = Comment.query.filter(Comment.news_id == news_id).order_by(Comment.create_time.desc()).all()
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='获取评论失败')

    # 6.1 用户点赞过的评论编号
    try:
        # 6.1.1 该用户点过所有的赞
        commentlikes = []
        if g.user:
            commentlikes = CommentLike.query.filter(CommentLike.user_id == g.user.id).all()

        # 6.1.2获取用户所有点赞过的评论编号
        mylike_comment_ids = []
        for commentLike in commentlikes:
            mylike_comment_ids.append(commentLike.comment_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg="获取点赞失败")

    #7.将评论的对象列表,转成字典列表
    comments_list = []
    for comment in comments:
        #将评论对象,转字典
        comm_dict = comment.to_dict()

        #添加is_like记录点赞
        comm_dict['is_like'] = False

        #判断用户是否有对评论点过赞
        if g.user and comment.id in mylike_comment_ids:
            comm_dict['is_like'] = True

        comments_list.append(comm_dict)



    #8.判断登陆的用户是否关注了新闻的作者
    is_followed = False
    if g.user and news.user:
        if g.user in news.user.followers:
            is_followed = True

    ##2.携带数据,渲染页面
    data = {
        'news_info':news.to_dict(),
        'user_info':g.user.to_dict() if g.user else "",
        'news_list':click_news_list,
        'is_collected':is_collected,
        'comments':comments_list,
        'is_followed':is_followed
    }

    return render_template('news/detail.html',data=data)